Fork me on GitHub

JAVA 类加载器

注意:所有文章除特别说明外,转载请注明出处.

类加载器

Java的类加载器就是在运行时在JVM中动态地加载所需的类,Java类加载器基于三个机制:委托、可见、单一。将classpath下的那些.class文件加载进内存,处理后成为字节码,这些工作是类加载器做的。

  • 委托机制: 指的是将加载类的请求传递给父加载器,如果父加载器找不到或者不能加载这个类,那么再加载它。
  • 可见性机制:指的是父加载器加载的类都能被子加载器看见,但是子加载器加载的类父加载器是看不见的。
  • 单一性机制:指的是一个类只能被同一种加载器加载一次。

    1.默认类加载器

    在系统中默认的类加载器有三个:BootStrap | ExtClassLoader | AppClassLoader 。

虽然说类加载器也是java类,但是BootStrap不是。验证程序:

public class ClassLoaderTest {
    public static void main(String[] args) {
        System.out.println(System.class.getClassLoader());
    }
}

output:

null

如果使用System.out.println(System.class.getClassLoader().toString);则报空指针异常(NullPointerException)。所以得到System类是由BootStrap类加载器加载的。

2.类加载器的委托机制

一般加载类的顺序:1.首先当前线程的类加载器去加载线程中的第一个类。2.然后如果类A应用了类B,java的虚拟机将使用加载类A的类加载器来加载类B。3.同时还可以直接调用ClassLoader.loadClass()方法来制定某个加载器去加载某个类。

3.自定义类加载器编写原理

父类:

loadClass(类加载的流程,模板)
findClass供子类覆盖的、被loadClass方法调用的类加载逻辑
defineClass得到class文件转换成字节码

子类:

覆盖findClass方法

如:
    protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
        synchronized (getClassLoadingLock(name)) {
        // First, check if the class has already been loaded
        Class<?> c = findLoadedClass(name);
        if (c == null) {
            long t0 = System.nanoTime();
            try {
                if (parent != null) {
                    c = parent.loadClass(name, false);
                } else {
                    c = findBootstrapClassOrNull(name);
                }
            } catch (ClassNotFoundException e) {
                // ClassNotFoundException thrown if class not found
                // from the non-null parent class loader
            }

            if (c == null) {
                // If still not found, then invoke findClass in order
                // to find the class.
                long t1 = System.nanoTime();
                c = findClass(name);

                // this is the defining class loader; record the stats
                sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                sun.misc.PerfCounter.getFindClasses().increment();
            }
        }
        if (resolve) {
            resolveClass(c);
        }
        return c;
    }
}

然后在API文档中的例子为:

class NetworkClassLoader extends ClassLoader {
     String host;
     int port;

     public Class findClass(String name) {
         byte[] b = loadClassData(name);
         return defineClass(name, b, 0, b.length);
     }

     private byte[] loadClassData(String name) {
         // load the class data from the connection
          . . .
     }
}

本文标题:JAVA 类加载器

文章作者:Bangjin-Hu

发布时间:2019年10月15日 - 09:22:26

最后更新:2020年03月30日 - 08:10:02

原始链接:http://bangjinhu.github.io/undefined/Java 类加载器/

许可协议: 署名-非商业性使用-禁止演绎 4.0 国际 转载请保留原文链接及作者。

Bangjin-Hu wechat
欢迎扫码关注微信公众号,订阅我的微信公众号.
坚持原创技术分享,您的支持是我创作的动力.